home *** CD-ROM | disk | FTP | other *** search
/ MacWorld 1999 March - Disc 1 / Macworld (1999-03) (Disk 1).dmg / Shareware World / Utilities / Text Processing / Alpha / Tcl / SystemCode / indentation.tcl < prev    next >
Encoding:
Text File  |  1998-11-21  |  16.1 KB  |  550 lines  |  [TEXT/ALFA]

  1. ## -*-Tcl-*-
  2.  # ###################################################################
  3.  #    Vince's    Additions -    an extension package for Alpha
  4.  # 
  5.  #    FILE: "indentation.tcl"
  6.  #                      created: 27/7/97 {1:08:08 am}    
  7.  #                     last update: 21/11/98 {3:41:21 pm}    
  8.  #    Author:    Vince Darley
  9.  #    E-mail:    <darley@fas.harvard.edu>
  10.  #      mail:    Division of    Applied    Sciences, Harvard University
  11.  #            Oxford Street, Cambridge MA    02138, USA
  12.  #       www:    <http://www.fas.harvard.edu/~darley/>
  13.  #    
  14.  # ###################################################################
  15.  ##
  16.  
  17. alpha::feature electricBraces 0.1 {global C C++ Java Tcl Perl} {
  18.     set electricBraces 0
  19. } {set electricBraces 1} {set electricBraces 0} help {
  20.     Enabling the 'Electric Braces' feature tells Alpha to treat the 
  21.     left or right brace '{', '}' keys as special keypresses which 
  22.     enter the '{' or '}' character, followed by a return and then 
  23.     indent the following line correctly.  It is useful for those 
  24.     programming modes in which '{' and '}' are used to delineate 
  25.     blocks of code in 'for' loops or 'if-then-else' groups etc.
  26. }
  27.  
  28. alpha::feature electricSemicolon 0.1 {global C C++ Java Perl} {
  29.     set electricSemicolon 0
  30. } {set electricSemicolon 1} {set electricSemicolon 0} help {
  31.     Enabling the 'Electric Semicolon' feature tells Alpha to treat the 
  32.     semicolon keys ';' as special keypresses which enters the ';' 
  33.     character followed by a return and then indents the following line 
  34.     correctly.  It is useful for some programming modes in which ';' 
  35.     normally ends a line.
  36.     
  37.     The ';' key is context-dependent so you can still enter a 
  38.     for( ; ; ) loop in C mode (for instace) without Alpha messing 
  39.     things up.
  40. }
  41.  
  42. alpha::feature electricReturn 0.1 {global} {
  43.     if {[info tclversion] >= 8.0} {
  44.     linkVar indentOnReturn
  45.     }
  46.     set indentOnReturn 0
  47. } {set indentOnReturn 1} {set indentOnReturn 0} help {
  48.     Enabling the 'Electric Return' feature tells Alpha to indent the 
  49.     following line automatically whenever you press return.
  50. }
  51.  
  52. alpha::feature electricColon 0.1 {global} {
  53.     set electricColon 0
  54. } {set electricColon 1} {set electricColon 0} help {
  55.     Enabling the 'Electric Colon' feature tells Alpha to carry out a 
  56.     special action when the user presses colon.
  57. }
  58.  
  59. alpha::feature autoContinueComment 0.1 {global} {
  60.     set autoContinueComment 0
  61. } {set autoContinueComment 1} {set autoContinueComment 0} help {
  62.     Enabling the 'autoContinueComment' feature tells Alpha to check when
  63.     the users hits return whether the current line is a comment, and if
  64.     so, to indent and insert comment characters so that the following
  65.     line continues the comment.
  66. }
  67.  
  68. alpha::feature indentUsingSpacesOnly 0.1 {global TeX} {
  69.     set indentUsingSpacesOnly 0
  70. } {set indentUsingSpacesOnly 1} {set indentUsingSpacesOnly 0} help {
  71.     If set, do not use tabs to indent, but spaces only.  This is mostly
  72.     useful for modes in which the 'tab' character has a special meaning,
  73.     such as python or TeX (the latter usually only for TeX as a programming
  74.     language, not as a document preparation system).
  75. }
  76.  
  77. alpha::feature commentsArentSpecialWhenIndenting 0.1 {global TeX} {
  78.     set commentsArentSpecialWhenIndenting 0
  79. } {set commentsArentSpecialWhenIndenting 1} {set commentsArentSpecialWhenIndenting 0} help {
  80.     Indent lines to level of previous line if set, otherwise to level 
  81.     of previous non-comment line (in which case Alpha will search 
  82.     backwards for some distance).  If you're in the habit of indenting 
  83.     your comments to the same level as your code, this setting 
  84.     shouldn't matter (and setting it is slightly more efficient).
  85.     
  86.     One case in which it can be _much_ more efficient is when your 
  87.     files contain vast comments (especially .dtx files in TeX mode, 
  88.     for instance).  For these files, you should activate this feature.
  89. }
  90.  
  91. namespace eval indent {}
  92. namespace eval Bind {}
  93. namespace eval text {}
  94.  
  95. proc IndentLine {} { bind::IndentLine }
  96.  
  97. proc typeText {t} {
  98.     if {[isSelection]} {
  99.     deleteSelection
  100.     }
  101.     insertText $t
  102. }
  103.  
  104. proc normalLeftBrace {} {
  105.     typeText "\{"
  106. }
  107. proc normalRightBrace {} {
  108.     typeText "\}"
  109.     blink [matchIt "\}" [pos::math [getPos] - 2]]
  110. }
  111.             
  112. proc literalChar {} {
  113.     return [expr {[lookAt [pos::math [getPos] - 1]] == "\\"}]
  114. }
  115.  
  116. # ◊◊◊◊ Electric indentation ◊◊◊◊ #
  117. proc bind::LeftBrace {} {
  118.     if {[isSelection]} { deleteSelection }
  119.     global electricBraces mode
  120.     if {!$electricBraces} {
  121.     insertText "\{"
  122.     return
  123.     }
  124.     mode::proc electricLeft
  125. }
  126.  
  127. proc ::electricLeft {} {
  128.     if {![catch {search -l [lineStart [pos::math [lineStart [getPos]] - 1]] \
  129.       -s -f 0 -r 0 "\}" [getPos]} res]} {
  130.     set end [getPos]
  131.     if {[pos::compare [getPos] != [maxPos]]} {
  132.         set end [pos::math $end + 1]
  133.     }
  134.     
  135.     if {[regexp "\}\[ \t\r\n\]*else" [getText [lindex $res 0] $end]]} {
  136.         set res2 [search -s -f 0 -r 1 {else} [getPos]]
  137.         oneSpace
  138.         set text [getText [lindex $res2 0] [getPos]]
  139.         if {[lookAt [pos::math [getPos] - 1]] != " "} {
  140.         append text " "
  141.         }
  142.         replaceText [pos::math [lindex $res 0] + 1] [getPos] " $text\{\r"
  143.         bind::IndentLine
  144.         return 
  145.     }
  146.     }
  147.     set pos [getPos]
  148.     set i [text::firstNonWsLinePos $pos]
  149.     
  150.     if {([pos::compare $i == $pos]) || ([lookAt [pos::math $pos - 1]] == " ")} {
  151.     insertText "\{\r" [text::indentString $pos] [text::Tab]
  152.     } else {
  153.     insertText " \{\r" [text::indentString $pos] [text::Tab]
  154.     }
  155. }
  156.  
  157. proc ::electricRight {} {
  158.     set pos [getPos]
  159.     set start [lineStart $pos]
  160.     
  161.     if {[catch {matchIt "\}" [pos::math $pos - 1]} matched]} {
  162.     beep
  163.     message "No matching '\{'!"
  164.     return
  165.     }
  166.     set text [getText [lineStart $matched] $matched]
  167.     regexp "^\[ \t\]*" $text indentation
  168.     if {[string trim [getText $start $pos]] != ""} {
  169.     insertText "\r" $indentation "\}\r" $indentation
  170.     blink $matched
  171.     return
  172.     }
  173.     set text "${indentation}\}\r$indentation"
  174.     replaceText $start $pos $text
  175.     goto [pos::math $start + [string length $text]]
  176.     blink [matchIt "\}" [pos::math $start - 2]]
  177. }
  178.  
  179. proc bind::RightBrace {} {
  180.     if {[isSelection]} { deleteSelection }
  181.     global electricBraces mode
  182.     if {!$electricBraces} {
  183.     insertText "\}"
  184.     catch {blink [matchIt "\}" [pos::math [getPos] - 2]]}
  185.     return
  186.     }
  187.     mode::proc electricRight
  188. }
  189.  
  190. proc bind::electricSemi {} {
  191.     if {[isSelection]} { deleteSelection }
  192.     global electricSemicolon mode
  193.     if {!$electricSemicolon} {
  194.     insertText ";"
  195.     return
  196.     }
  197.     mode::proc electricSemi
  198. }
  199.  
  200. proc ::electricSemi {} {
  201.     set pos [getPos]
  202.     set start [lineStart $pos]
  203.     set text [getText $start $pos]
  204.     
  205.     if {[string first "for" $text] != "-1"} {
  206.     set paren 0
  207.     set len [string length $text]
  208.     for {set i 0} {$i < $len} {incr i} {
  209.         switch -- [string index $text $i] {
  210.         "("    { incr paren }
  211.         ")"    { incr paren -1 }
  212.         }
  213.     }
  214.     if {$paren != 0} {
  215.         insertText ";"
  216.         return
  217.     }
  218.     }
  219.     
  220.     insertText ";\r" [text::indentString $pos]
  221. }
  222.  
  223. ## 
  224.  # -------------------------------------------------------------------------
  225.  #     
  226.  # "bind::CarriageReturn" --
  227.  #    
  228.  #    General    purpose    CR procedure.  Should be bound to 'return' for all 
  229.  #    modes really.  Calls a mode-specific procedure if required.
  230.  # -------------------------------------------------------------------------
  231.  ##
  232. proc bind::CarriageReturn {} {
  233.     if {[isSelection]} { deleteSelection }
  234.     global autoContinueComment
  235.     if {$autoContinueComment && ([text::isInComment [set p [getPos]] start])} {
  236.     # special case for beginning of line
  237.     if {[pos::compare $p == [lineStart $p]]} {
  238.         backwardChar
  239.     }
  240.     insertText "\r${start}"
  241.     return
  242.     }
  243.     mode::proc carriageReturn
  244. }
  245.  
  246. proc ::carriageReturn {} {
  247.     insertText "\r"
  248.     global indentOnReturn
  249.     if {$indentOnReturn} {bind::IndentLine}
  250. }
  251.  
  252. proc bind::IndentLine {} {
  253.     mode::proc indentLine
  254. }
  255.  
  256. proc insertActualTab {} { typeText "\t" }
  257.  
  258.  
  259.  
  260. ## 
  261.  # -------------------------------------------------------------------------
  262.  #     
  263.  # "text::isInComment" --
  264.  #    
  265.  #    Are    we in a    block comment?    Just checks    if both    the    given line and the
  266.  #    next line commence with    any    of a set of    known block-comment    characters.
  267.  #    Not 100% satisfactory for C comments, but fine for all others.
  268.  # -------------------------------------------------------------------------
  269.  ##
  270. proc text::isInComment {pos {st ""}} {
  271.     set p [lineStart $pos]
  272.     if {[pos::compare $pos == $p] && [pos::compare $p != [minPos]]} { 
  273.     set pos [pos::math $pos - 1] ; set p [lineStart $pos] 
  274.     }
  275.     set q [nextLineStart $pos]
  276.     set t [getText $p $q]
  277.     if { $st != "" } {
  278.     upvar $st a
  279.     }
  280.     if {![catch {commentCharacters "Paragraph"} cpar]} {
  281.     if {[regexp "^\[ \t\]*[quote::Regfind [lindex $cpar 0]]" $t a]} {
  282.         if {![regexp "[quote::Regfind [lindex $cpar 1]]" $t]} {
  283.         set len [string length [lindex $cpar 2]]
  284.         set a [string range $a 0 [expr {[string length $a] - $len -1}]]
  285.         append a [lindex $cpar 2]
  286.         return 1
  287.         }
  288.     }
  289.     }
  290.     # if the next line is a comment 
  291.     set qq [text::firstNonWsLinePos $q]
  292.     if {[pos::compare $qq == [maxPos]]} { 
  293.     return 0 
  294.     }
  295.     foreach commentCh [commentCharacters "General"] {    
  296.     if {[regexp  "^\[ \t\]*[quote::Regfind ${commentCh}]\[ \t\]*" $t a]} {
  297.         # if we hit return in the middle of a line
  298.         if {[string trim [getText $pos $q]] != "" && [pos::compare $pos != $p]} { 
  299.         return 1
  300.         }
  301.         if {[getText $qq [pos::math $qq + [string length $commentCh]]] == $commentCh} {
  302.         return 1
  303.         }
  304.     }
  305.     }
  306.     return 0
  307. }
  308.  
  309.  
  310. # ◊◊◊◊ Indentation utility routines ◊◊◊◊ #
  311.  
  312. proc posX {pos} {return [lindex [posToRowCol $pos] 1] }
  313. # the above version doesn't work!
  314. if {[info tclversion] < 8.0} {
  315. proc posX {pos} {return [string length [text::maxSpaceForm [getText [lineStart $pos] $pos]]]}
  316. }
  317.  
  318. proc text::firstNonWs {pos} {
  319.     set p [text::firstNonWsPos $pos]
  320.     if {[pos::compare $p > [minPos]]} {
  321.     return [lookAt $p]
  322.     } else {
  323.     return ""
  324.     }
  325. }
  326.  
  327. ## 
  328.  # -------------------------------------------------------------------------
  329.  #   
  330.  # "text::firstNonWsPos" --
  331.  #  
  332.  #  This returns the position of the first non-whitespace character from
  333.  #  the start of pos' line.  It need not return something on the same
  334.  #  line.
  335.  # -------------------------------------------------------------------------
  336.  ##
  337. proc text::firstNonWsPos {pos} {
  338.     if {[catch {lindex [search -s -f 1 -r 1 "\[^ \t\r\n\]" [lineStart $pos]] 0} res]} {
  339.     return [lineStart $pos]
  340.     } else {
  341.     return $res
  342.     }
  343. }
  344.  
  345. proc text::firstNonWsLinePos {pos} {
  346.     if {[catch {lindex [search -s -f 1 -r 1 "\[^ \t\]" [lineStart $pos]] 0} res]} {
  347.     return [lineStart $pos]
  348.     } else {
  349.     return $res
  350.     }
  351. }
  352.  
  353. proc text::indentation {pos} {
  354.     return [search -s -m 0 -f 1 -r 1 "^\[ \t\]*\[^ \t\]" [lineStart $pos]]
  355. }
  356.  
  357. ## 
  358.  # -------------------------------------------------------------------------
  359.  # 
  360.  # "text::minSpaceForm" --
  361.  # 
  362.  #  Converts to minimal form: tabs then spaces.  Uses one regsub to do
  363.  #  the job.  Note that the regexp used relies upon the left-to-right
  364.  #  priority of branch matching.  If the regexp library used is more
  365.  #  sophisticated and finds maximal matches, then this is no good.
  366.  #  In that case use:
  367.  #        regsub -all $sp $ws "\t" ws
  368.  #        regsub -all " +\t" $ws "\t" ws
  369.  # -------------------------------------------------------------------------
  370.  ##
  371. proc text::minSpaceForm {ws} {
  372.     regsub -all "([spacesEqualTab]| +\t)" $ws "\t" ws
  373.     return $ws
  374. }
  375.  
  376. ## 
  377.  # -------------------------------------------------------------------------
  378.  # 
  379.  # "text::maxSpaceForm" --
  380.  # 
  381.  #  Converts it to maximal form - just spaces.
  382.  #  Just uses one funky regsub to do the job!  Takes account of tab-size,
  383.  #  spaces interspersed with tabs,...
  384.  # -------------------------------------------------------------------------
  385.  ##
  386. proc text::maxSpaceForm {ws} {
  387.     set sp [spacesEqualTab]
  388.     regsub -all "(($sp)*) *\t" $ws "\\1$sp" ws
  389.     return $ws
  390. }
  391.  
  392. ## 
  393.  # -------------------------------------------------------------------------
  394.  # 
  395.  # "spacesEqualTab" --
  396.  # 
  397.  #  Return the number of spaces equivalent to a single tab.
  398.  # -------------------------------------------------------------------------
  399.  ##
  400. proc spacesEqualTab {} {
  401.     getWinInfo a
  402.     string range "              " 1 $a(tabsize)
  403. }
  404.  
  405. proc doubleLookAt {pos} {return [getText $pos [pos::math $pos + 2]]}
  406.  
  407. set bind::_IndentSpaces "                                                   \
  408.                                          "
  409. set bind::_IndentTabs "\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t"
  410.  
  411. proc text::indentOf {size} {
  412.     global bind::_IndentSpaces bind::_IndentTabs indentUsingSpacesOnly
  413.     if {$indentUsingSpacesOnly} {
  414.     return [string range ${bind::_IndentSpaces} 1 $size]
  415.     } else {
  416.     getWinInfo a
  417.     set ret [string range ${bind::_IndentTabs} 1 [expr $size / $a(tabsize)]]
  418.     append ret [string range ${bind::_IndentSpaces} 1 [expr $size % $a(tabsize)]]
  419.     }
  420.     return $ret
  421. }
  422.  
  423. # returns the indent string of the line named by 'pos'
  424. proc text::indentString {pos} {
  425.     set beg [lineStart $pos]
  426.     regexp "^\[ \t\]*" [getText $beg [nextLineStart $beg]] white
  427.     return $white
  428. }
  429.  
  430. # returns the indent string of the line up to position 'pos' 
  431. proc text::indentTo {pos} {
  432.     regexp "^\[ \t\]*" [getText [lineStart $pos] $pos] white
  433.     return $white
  434. }
  435.  
  436. proc text::halfTab {} {
  437.     global indent_amounts
  438.     return [string range "              " 1 $indent_amounts(1)]
  439. }
  440. proc text::Tab {} {
  441.     global indentationAmount
  442.     return [text::indentOf $indentationAmount]
  443. }
  444.  
  445. proc text::getTabSize {} {
  446.     getWinInfo a
  447.     return $a(tabsize)
  448. }
  449.  
  450. # ◊◊◊◊ General purpose indentation ◊◊◊◊ #
  451.  
  452. proc indentSelection {} {
  453.     mode::proc indentRegion
  454. }
  455.  
  456. ## 
  457.  # -------------------------------------------------------------------------
  458.  # 
  459.  # "text::inCommentBlock" --
  460.  # 
  461.  #  Returns 'startpos endpos' if true, else returns an error.  Not 
  462.  #  particularly robust, but not too bad either
  463.  # -------------------------------------------------------------------------
  464.  ##
  465. proc text::inCommentBlock {pos} {
  466.     set chars [commentCharacters Paragraph]
  467.     set start [string trim [lindex $chars 0]]
  468.     set end [string trim [lindex $chars 1]]
  469.     if {$start == $end} {
  470.     error "No"
  471.     }
  472.     set cS [search -s -f 0 -r 0 -l [pos::math $pos - 1000] $start $pos]
  473.     set cE [search -s -f 1 -r 0 -l [pos::math $pos + 1000] $end [lindex $cS 1]]
  474.     if {[pos::compare $pos >= [lindex $cE 1]]} {    
  475.     error "No"
  476.     } else {
  477.     return [list [lindex $cS 0] [lindex $cE 1]]
  478.     }
  479. }
  480.  
  481.  
  482. # Tom's new regexp which I don't use now.  Shame.
  483. #set commentRegexp       {/\*[^*]*\*+([^/*][^*]*\*+)*/}
  484.  
  485. #########################################################################
  486. # Generic C-style indentation (works for Tcl and Perl)
  487. # Significant changes by Vince.
  488. proc ::indentLine {} {
  489.     global commentsArentSpecialWhenIndenting
  490.     # get details of current line
  491.     set beg [lineStart [getPos]]
  492.     set text [getText $beg [nextLineStart $beg]]
  493.     regexp "^\[ \t\]*" $text white
  494.     set len [string length $white]
  495.     set epos [pos::math $beg + $len]
  496.  
  497.     if {[pos::compare $beg != [minPos]]} {
  498.     # Find last previous non-comment line and get its leading whitespace
  499.     set pos $beg
  500.     while 1 {
  501.         if {[pos::compare $pos == [minPos]] || [catch {search -s -f 0 -r 1 -i 0 -m 0 "^\[ \t\]*\[^ \t\r\n\]" [pos::math $pos - 1]} lst]} {
  502.         # search failed at top of file
  503.         set line "#"
  504.         set lwhite 0
  505.         break
  506.         }
  507.         if {!$commentsArentSpecialWhenIndenting && \
  508.           ![catch {text::inCommentBlock [lindex $lst 0]} res]} {
  509.         set pos [lindex $res 0]
  510.         } else {
  511.         set line [getText [lindex $lst 0] [pos::math [nextLineStart [lindex $lst 0]] - 1]]
  512.         set lwhite [posX [pos::math [lindex $lst 1] - 1]]    
  513.         break
  514.         }
  515.     }
  516.     
  517.     regexp "(\[^ \t\])\[ \t\]*\$" $line "" nextC
  518.     global indentationAmount electricColon
  519.     if {($nextC == "\{")} {
  520.         incr lwhite $indentationAmount
  521.     } elseif {$nextC == ":" && $electricColon} {
  522.         incr lwhite [expr {$indentationAmount /2}]
  523.     }
  524.     
  525.     if {[regexp ":\[ \t\r\n\]*\$" $text] && $electricColon} {incr lwhite [expr {-$indentationAmount / 2}]}
  526.     if {[lookAt $epos] == "\}"} {
  527.         incr lwhite [expr {-$indentationAmount}]
  528.     }
  529.     } else {
  530.     set lwhite 0
  531.     }
  532.     set lwhite [text::indentOf $lwhite]
  533.     if {$white != $lwhite} {
  534.     replaceText $beg $epos $lwhite
  535.     }
  536.     goto [pos::math $beg + [string length $lwhite]]
  537. }
  538.  
  539.  
  540. proc ::indentRegion {} {
  541.     set from [lindex [posToRowCol [getPos]] 0]
  542.     set to [lindex [posToRowCol [selEnd]] 0]
  543.     select [getPos]
  544.     while {$from <= $to} {
  545.     goto [rowColToPos $from 0]
  546.     bind::IndentLine
  547.     incr from
  548.     }
  549. }
  550.